# DISEÑO HARDWARE CON VHDL

- ✓ Lenguaje de Descripción Hardware
- ✓ Descripción Hardware en VHDL
- ✓ Estructura de un Modelo VHDL
- √ Elementos Básicos de VHDL
- √ Ejemplo: Máquina de Estados Finita

Javier Resano, GAZ: Grupo de Arquitectura de Computadores



Departamento de Informática e Ingeniería de Sistemas

- ¿Qué es HDL?
  - Lenguajes creados para el diseño de circuitos:
    - » Nivel de puerta (gate level).
    - » Nivel de comportamiento (behavioural level).
  - La estructura del lenguaje sugiere el diseño hardware.
- ¿Por qué usar HDL?
  - El diseño es mucho más rápido.
  - Permite simular antes de la implementación física.
  - Las herramientas de diseño se encargan de los pasos automatizables.

No es lo mismo describir **hardware** que programar **software** 

- VHDL
  - VHSIC (Gobierno de EE.UU. 1980).
  - IEEE VHDL'87.
  - <u>www.vhdl.org</u>
- Verilog
  - Desarrollado por CADENCE.
  - IEEE 1364.
  - <u>www.eda.org</u>
- A partir de C:
  - Handel C
  - System C
  - C/C++ (síntesis de alto nivel)

#### • VHDL

- Descripción de la estructura del circuito
  - Descomposición en sub-circuitos
  - Interconexión de sub-circuitos
  - Comportamiento
  - Estructural
- Permite la especificación de la funcionalidad de un circuito utilizando formas familiares en los lenguajes de programación
- Permite la simulación del circuito antes de su fabricación
  - Testear y comparar alternativas sin necesidad de prototipos hardware

### • Verilog:

- Circuitos Integrados + Sistemas Analógicos + Señales (comportamiento o estructural)
- Analógico:
  - Bloques Analógicos
  - Operadores Integrales y Derivada
  - Transformada de Laplace
- Señales:
  - Módulo de señales Analógico/Digital
  - Reglas para convolución de señales
  - Conversores A-D y D-A

- ¿ Por qué VHDL?
  - Lenguaje más utilizado para el diseño de CI
  - Gran cantidad de compiladores de eficacia probada para su implementación en FPGA o sobre silicio.
     (Synopsis, Xilinx, Mentor Graphics, Cadence, hp ...)
  - Verilog necesita descripciones a un nivel más bajo. Es más utilizado para diseñar circuitos de tratamiento de señales. Es el más utilizado en América.

## Descripción Hardware en VHDL

• Descripción de un Sistema Digital:

Un sistema digital está descrito por sus entradas y sus salidas, donde las salidas dependen de las entradas



#### Estructura de un Modelo VHDL

### Los modelos VHDL están formado por 2 partes:

#### • Entity

- Define externamente al circuito o subcircuito
- Nombre y número de puertos, tipos de datos de entrada y salida
- Tienes toda la información necesaria para conectar tu circuito a otros circuitos

#### • Architecture

- Define internamente el circuito
- Señales internas, funciones, procedimientos, constantes ...
- Puede haber varias arquitecturas para una entidad

```
entity F is

port (A, B: in bit; Y out bit);
end F

architecture circuito of F is
signal D, E: bit_vector(1 downto 0);
signal H: bit;
begin
(...)
end architecture circuito;

Entradas y salidas

Cada señal es un cable
funcionalidad

funcionalidad
```

```
-- comment

ENTITY name IS

GENERIC (parameter list);

PORT (io list);

DIN IN: Bit_vector(Size-a downto 0)

END name;
```

```
parameter list:
  parameter_i : type_name; ...

retardo: time; error: boolean

io list:
  port_i : io_type type_name; ...

io_type
IN | OUT | INOUT | BUFFER
```

entradaA: in bit\_vector(7 downto 0);
salidaC: out std\_logic;

```
type_name
BIT {'0','1'} – STD_LOGIC
BIT_VECTOR (range) – STD_LOGIC_VECTOR (range)
BOOLEAN {TRUE,FALSE}
CHARACTER {ascii}
STRING {ascii}
SEVERITY_LEVEL {NOTE, WARNING, ERROR, FALURE}
INTEGER [range]
NATURAL [range]
POSITIVE [range]
REAL [range]
TIME
                                                        1000
                                                    0 TO 3 ->>> Es un 1
range
                                                   3 DOWNTO ->>> Es un 8
n min TO n max n max DOWNTO n min
```

#### • Señales:

- Representan elementos de memoria o conexiones
- Los puertos de una entidad
- En la arquitectura antes del BEGIN, lo cual nos permite realizar conexiones entre diferentes módulos

señal <= valor

#### Variables

- Se utilizan como índices (instrucciones de bucle o modelar componentes)
- Las variables NO representan conexiones o estados de memoria

variable := valor

## No las vamos a usar

• Descripcion funcional:

```
ARCHTECTURE arch_name OF entity_name IS
signal_i;
BEGIN
-- sentencias concurrentes
 AND, OR, NOT, NAND, XOR, +, - ...
 WAIT
-- procedural (sequencial ) assertions
PROCESS (sensitivity list)
variable_j;
BEGIN
-- asignaciones, sentencias condicionales, bucles
END PROCESS;
END arch_name;
```

- Dos métodos para representar la funcionalidad:
  - Processes:
    - Conjunto de sentencias secuenciales. El orden de las sentencias puede afectar al resultado.
  - Sentencias concurrentes:
    - Sentencias e invocaciones a procesos que se ejecutan en paralelo. Da igual en qué orden se escriban.

```
Architecture ver1 of test is begin
```

```
c <= a and b;
z <= c when e ='1' else 'Z';
seq: process (a, b)
```

Sentencias concurrentes

```
begin
```

end ver1;

```
if a = b then
    f <= a or b;
end if;
If a = "1000" then
    d <= f;
end if;
end process;</pre>
```

Las entradas de las sentencias tienen que estar en la lista de sensibilidad

Sentencias secuenciales: Sólo dentro de los procesos

• Operadores basicos:

```
abs
*, /, mod, rem
+ (sig.), - (sig)
+, -, &
and, or, nand, nor ...
```

```
y<=(x1 and x2) or d(0);
y(1) <= x1 and not x2;

y: bit_vector (1 downto 0);
y <= x1&x2;</pre>
```

## • Sentencias Concurrentes (I):

```
signal_name <= valor_1 when condición1 else
valor_2 when condición2 else
...
valor_i when condicióni else
otro_valor;
```



```
salida <= "00" when entrada = "0001" else

"01" when entrada = "0010" else

"10" when entrada = "0100" else

"11";
```

### • Sentencias Concurrentes (II):



• Sentencias secuenciales condicionales (I):

• Sentencias secuenciales condicionales (II):

```
case expression is
   when choice_l => ... --seq. statements;
   when choice_n => ... --seq. statements;
   when others => ... --seq. statements;
end case;
```

```
case RGB is
when "111" => r <= 0;
when "100" => r <= 1;
when "110" => r <= 1;
when others => r <= 0;
end case;
```

# Síntesis de Sentencias Secuenciales

## • Simulación de procesos:

- Un proceso se simula si cambia alguna señal de su lista de sensibilidad
- Si un proceso representa lógica combinacional su lista debería incluir todas las entradas
- Si un proceso representa lógica secuencial (un registro o un contador), su lista sólo debe incluir la señal de reloj y el reset o clear en caso de que sea asíncrono

# Síntesis de Sentencias Secuenciales

• Implementación de procesos:



# Síntesis de Sentencias Secuenciales

• Implementación de procesos: registro de 32 bit

```
Reg: process (clk)
 begin
      si ha habido flanco de subida
   if (clk'event and clk = '1') then
    if (reset = '1') then
      else
      if (load='1') then
        Dout <= Din;
      end if;
    end if;
   end if;
 end process;
```

```
library ieee;
use ieee.std logic 1164.all;
entity encoder is
    port (invec: in std logic vector(7 downto 0); enc out: out std logic vector(2 downto 0));
end encoder;
architecture rtl of encoder is
begin
 encode: process (invec)
begin
  case invec is
   when "00000001" =>
                           enc out <= "000";
                           enc out <= "001";
   when "00000010" =>
   when "00000100" =>
                           enc out <= "010";
   when "00001000" =>
                           enc out <= "011";
   when "00010000" =>
                           enc_out <= "100";
   when "00100000" =>
                          enc out <= "101";
   when "01000000" => enc out <= "110":
   when "10000000" =>
                           enc out <= "111";
   when others => enc out \leq "000":
   end case;
  end process;
end rtl:
```

```
library ieee;
use ieee.std logic 1164.all;
entity encoder is
 port (invec:in std_logic_vector(7 downto 0); enc_out:out_std_logic_vector(2 downto 0));
end encoder;
architecture rtl of encoder is
begin
 process (invec)
  begin
   if invec(7) = '1' then
                            enc out <= "111";
   elsif invec(6) = '1' then
                               enc out <= "110";
   elsif invec(5) = '1' then
                               enc out <= "101";
   elsif invec(4) = '1' then
                               enc out <= "100";
   elsif invec(3) = '1' then
                               enc out <= "011";
   elsif invec(2) = '1' then
                               enc out <= "010";
   elsif invec(1) = '1' then
                               enc out <= "001";
   elsif invec(0) = '1' then
                               enc out <= "000";
            enc out <= "000";
   else
   end if;
   end process:
end rtl;
```

```
library ieee;
use ieee.std_logic_1164.all;
entity encoder is
port (invec: in std_logic_vector(7 downto 0); enc_out: out std_logic_vector(2 downto 0));
end encoder;
architecture rtl of encoder is
begin
 enc_out <= "111" when invec(7) = '1' else
        "110" when invec(6) = '1' else
        "101" when invec(5) = '1' else
        "100" when invec(4) = '1' else
        "011" when invec(3) = '1' else
        "010" when invec(2) = '1' else
        "001" when invec(1) = '1' else
        "000" when invec(0) = '1' else
        "000":
end rtl:
```

• Bucles (sólo dentro de los procesos):

```
for loop_var in range loop
... --sequential statements
end loop;
```



for i in 3 downto 0 loop
-- i es una variable q no necesita
-- ser declarada
 d(i) <= a(i+1);
end loop;</pre>

```
while condición loop... --sequential statementsend loop;
```



while contador > 0 loop
contador := contador-1;
resultado := resultado+data;
end loop;

### No las vamos a usar

• Expresiones secuenciales:

wait;
wait on signal\_list;
wait for time\_expression;
wait until condition;

Para esperar 10ns en una simulación wait for 10ns;

signal'event signal'last\_event signal'last\_value Para que un registro haga algo sólo en el flanco de subida:

if clock'event and clock = '1';

- Descripción estructural: (estructural)
  - Describe las interconexiones entre distintos módulos
  - Estos módulos pueden a su vez tener un modelo estructural o de comportamiento
  - Normalmente esta es una descripción a más alto nivel

• Descripción Estructural:

```
ARCHTECTURE arch_name OF entity_name IS
    COMPONENT component_name
        GENERIC (...);
    PORT (...);
    END COMPONENT;
signal declarations;
BEGIN
    component_i: component_name
        GENERIC MAP (parameter_value)
        PORT MAP (io_name)
END arch_name;
```

```
» Ejemplo
                                                mod G
architecture structure of F is
   component G
                                                     Yg
       port (Ag, Bg: in bit; Yg: out bit);
                                                                  mod I
   end component;
                                                             YA
                                               Bg
                                                                           Yout
                                                                  Ai T
   component H
                                                mod H
       port (Ah, Bh: in bit; Yh: out bit);
                                               ► Ah
                                                                  Bi
   end component;
                                                              YB
                                                  H Yh
   component |
       port (Ai, Bi: in bit;Yi: out bit);
                                                Bh
   end component;
   signal YA, YB, Yout: bit;
begin
   mod_G: G port map (Ag=>A; Bg=>B; Yg=>YA);
   mod_H: H port map (Ah=>A; Bh=>B; Yh=>YB);
   mod_I : I port map (Ai=>YA; Bi=>YB; Yi=>Yout);
   Y<=Yout;
end structure;
```

• **GENERATE**: Las secuencias de generación de componentes permiten crear una o más copias de un conjunto de interconexiones, lo cual facilita el diseño de circuitos mediante descripciones estructurales.

for indice in rango generate-- instrucciones concurrentesend generate;

```
component comp
   port( x: in bit; y: out bit);
end comp
...
signal a, b: bit_vector(0 to 7)
...
gen: for i in 0 to 7 generate
   u: comp port map (a(i),b(i));
end generate gen;
```

### No lo vamos a usar

• Todo circuito secuencial se divide en un circuito combinacional que implementa la salida del circuito y la transición al siguiente estado y en unos elementos de almacenamiento.



architecture comportamiento of FSM is internal signal declarations; begin

síncrono: **process** ( *clock, reset*) **begin** *vhdl indicando los flip-flops* **end process** síncrono;

comb: **process** ( *sensitivity list*) **begin** *vhdl indicando la lógica combinacional* **end process** comb;

end comportamiento;



```
library IEEE;
use IEEE.std_logic_1164.all;
entity FSM is
     port(reset, E, clk: in bit; O: out bit);
end FSM:
architecture ARCH of FSM is
type ESTADOS is (S1, S2, S3,S4);
signal ESTADO, SIG ESTADO: ESTADOS;
begin
 SINCRONO: process(clk,reset)
 begin
     if reset ='1' then
              ESTADO<=S1;
     elsif clk'event and clk='1' then
              ESTADO<= SIG ESTADO;
      end if:
 end process SINCRONO;
```

```
COMB: process(ESTADO,E)
begin
     case ESTADO is
     when S1 =>
             O \le '0':
             if (E='0') then SIG_ESTADO<=S2;
             else SIG ESTADO<=S1;</pre>
             end if:
     when S2 =>
             O \le '0';
             if (E='0') then SIG ESTADO<=S3;
             else SIG ESTADO<=S1;</pre>
             end if:
     when S3 =>
             O \le '0';
             if (E='0') then SIG ESTADO<=S3;
             else SIG_ESTADO<=S4;</pre>
             end if:
     when S4 =>
             O <= '1';
             if (E='0') then SIG ESTADO<=S2;
             else SIG ESTADO<=S1;</pre>
             end if:
end case:
end process control;
```

```
library IEEE;
use IEEE.std_logic_1164.all;
entity FSM is
     port(reset, E, clk: in bit; O: out bit);
end FSM:
architecture ARCH of FSM is
type ESTADOS is (S1, S2, S3,S4);
signal ESTADO, SIG ESTADO: ESTADOS;
begin
 SINCRONO: process(clk,reset)
 begin
     if reset ='1' then
              ESTADO<=S1;
     elsif clk'event and clk='1' then
              ESTADO<= SIG ESTADO;
      end if:
 end process SINCRONO;
```

```
COMB: process(ESTADO,E)
begin
     case ESTADO is
     when S1 =>
             O \le '0':
             if (E='0') then SIG_ESTADO<=S2;
             else SIG ESTADO<=S1;</pre>
             end if:
     when S2 =>
             O \le '0';
             if (E='0') then SIG ESTADO<=S3;
             else SIG ESTADO<=S1;</pre>
             end if:
     when S3 =>
             O \le '0';
             if (E='0') then SIG ESTADO<=S3;
             else SIG_ESTADO<=S4;</pre>
             end if:
     when S4 =>
             O <= '1';
             if (E='0') then SIG ESTADO<=S2;
             else SIG ESTADO<=S1;</pre>
             end if:
end case:
end process control;
```